﻿using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Windows;

namespace LuminaDesk.Core.Views.Controls;

public class CompoundSlider : ShadowSlider
{
	#region Type

	private class Item(object source)
	{
		/// <summary>
		/// Binding source
		/// </summary>
		public object Source { get; set; } = source;

		public List<CompoundSlider> Sliders { get; } = [];
	}

	private class ItemHolder
	{
		private readonly List<Item> _items = [];

		public void Add(object source, CompoundSlider slider)
		{
			if (source is null)
				return;

			var item = _items.FirstOrDefault(x => ReferenceEquals(x.Source, source));
			if (item is null)
			{
				item = new Item(source);
				_items.Add(item);
			}
			else if (item.Sliders.Any(x => ReferenceEquals(x, slider)))
				return;

			item.Sliders.Add(slider);
		}

		public void Remove(object source, CompoundSlider slider)
		{
			if (source is null)
				return;

			var item = _items.FirstOrDefault(x => ReferenceEquals(x.Source, source));
			if (item is null)
				return;

			item.Sliders.Remove(slider);
			if (item.Sliders.Any())
				return;

			item.Source = null;
			_items.Remove(item);
		}

		public bool TryGetItem(object source, out Item item)
		{
			if (source is null)
				item = null;
			else
				item = _items.FirstOrDefault(x => ReferenceEquals(x.Source, source));

			return (item is not null);
		}
	}

	#endregion

	private static readonly ItemHolder _holder = new();

	/// <summary>
	/// Binding source bound to this instance
	/// </summary>
	private object _source;

	public override void OnApplyTemplate()
	{
		base.OnApplyTemplate();

		if (DesignerProperties.GetIsInDesignMode(this))
			return;

		this.Loaded += (_, _) =>
		{
			_source = this.DataContext;
			_holder.Add(_source, this);
		};

		this.Unloaded += (_, _) =>
		{
			_holder.Remove(_source, this);
			_source = null;
		};
	}

	#region Unison

	private static event EventHandler<(object source, double level, bool update)> Moved; // Static event

	public bool IsUnison
	{
		get { return (bool)GetValue(IsUnisonProperty); }
		set { SetValue(IsUnisonProperty, value); }
	}
	public static readonly DependencyProperty IsUnisonProperty =
		DependencyProperty.Register(
			"IsUnison",
			typeof(bool),
			typeof(CompoundSlider),
			new PropertyMetadata(
				false,
				(d, e) =>
				{
					var instance = (CompoundSlider)d;

					if ((bool)e.NewValue)
					{
						Moved += instance.OnMoved;
					}
					else
					{
						Moved -= instance.OnMoved;
					}
				}));

	public int ValueUnison
	{
		get { return (int)GetValue(ValueUnisonProperty); }
		set { SetValue(ValueUnisonProperty, value); }
	}
	public static readonly DependencyProperty ValueUnisonProperty =
		DependencyProperty.Register(
			"ValueUnison",
			typeof(int),
			typeof(CompoundSlider),
			new PropertyMetadata(
				0,
				(d, e) =>
				{
					var instance = (CompoundSlider)d;

					if (instance.IsUnison)
					{
						if (instance.IsFocused)
						{
							// This route is handled by OnValueChanged method.
							instance._update = true;
						}
						else if ((instance._source is not null)
							&& instance.TryGetLevel((int)e.NewValue, out double level))
						{
							Moved?.Invoke(instance, (instance._source, level, update: true));
						}
					}
				}));

	private bool _update;

	protected override void OnValueChanged(double oldValue, double newValue)
	{
		base.OnValueChanged(oldValue, newValue);

		var update = _update;
		_update = false;

		if (IsUnison && this.IsFocused && (_source is not null)
			&& this.TryGetLevel(out double level))
		{
			Moved?.Invoke(this, (_source, level, update: update));
		}
	}

	private void OnMoved(object sender, (object source, double level, bool update) e)
	{
		if (ReferenceEquals(this, sender) || ReferenceEquals(this._source, e.source))
			return;

		if (!_holder.TryGetItem(_source, out Item item) ||
			!ReferenceEquals(item.Sliders.FirstOrDefault(x => x.IsUnison), this))
			return;

		if (e is { level: >= 0 })
		{
			SetLevel(e.level);
		}
		else if (e is { update: true })
		{
			base.EnsureUpdateSource();
		}
	}

	public override void EnsureUpdateSource()
	{
		// Ensure the binding source bound to this instance is updated.
		base.EnsureUpdateSource();

		// Ensure the binding sources bound to other instances are updated.
		if (_source is not null)
		{
			Moved?.Invoke(this, (_source, -1, update: true));
		}
	}

	#endregion
}